3

defer会经常用到,但是这个坑对于我并不会踩到,因为我通常不会用到命名返回参数,一则没有多少必要,二则增加了代码阅读难度。不过,这个坑能让人很好地理解return这个关键字,所以在此记录下。

func test() (res int) {
    res = 1
    defer func() {
        res++
    }()
    return 0
}

默默在心里运行这个程序,第一个返回值应该是0嘛,其实不然,return并不是个原子操作。分为了,赋值和返回两个操作,<return 0>其实是等价于<res = 0, return res>,而defer是在赋值操作后执行的,所以这个函数等价于:

func test() (res int) {
    res = 1    
    res = 0    //赋值
    func() {
        res++    //defer
    }()
    return res //返回
}

再来看个例子:

 func test() (res int) {
    tmp := 1
    defer func() {
        tmp++
    }()
    return tmp
}

按刚才的经验分析下这个程序,tmp=1,然后返回前又加了1,应该是2吧。其实又错了,注意返回值命名是res,而不是tmp,defer改变的只能是tmp。这个函数可以等价于:

 func test() (res int) {
    tmp := 1
    res = tmp
    func() {
        tmp++
    }()
    return 
}

结果应该是1。


Cedrus
363 声望9 粉丝

不积跬步,无以致千里